/************************************************************************
 * This file is part of EspoCRM.
 *
 * EspoCRM – Open Source CRM application.
 * Copyright (C) 2014-2024 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
 * Website: https://www.espocrm.com
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Affero General Public License for more details.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <https://www.gnu.org/licenses/>.
 *
 * The interactive user interfaces in modified source and object code versions
 * of this program must display Appropriate Legal Notices, as required under
 * Section 5 of the GNU Affero General Public License version 3.
 *
 * In accordance with Section 7(b) of the GNU Affero General Public License version 3,
 * these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
 ************************************************************************/

define('views/email-folder/list-side', ['view'], function (Dep) {

    return Dep.extend({

        template: 'email-folder/list-side',

        FOLDER_ALL: 'all',
        FOLDER_INBOX: 'inbox',
        FOLDER_DRAFTS: 'drafts',

        events: {
            'click [data-action="selectFolder"]': function (e) {
                e.preventDefault();

                let id = $(e.currentTarget).data('id');

                this.actionSelectFolder(id);
            }
        },

        data: function () {
            let data = {};

            data.selectedFolderId = this.selectedFolderId;
            data.showEditLink = this.options.showEditLink;
            data.scope = this.scope;

            return data;
        },

        actionSelectFolder: function (id) {
            this.$el.find('li.selected').removeClass('selected');

            this.selectFolder(id);

            this.$el.find('li[data-id="'+id+'"]').addClass('selected');
        },

        setup: function () {
            this.scope = 'EmailFolder';
            this.selectedFolderId = this.options.selectedFolderId || this.FOLDER_ALL;
            this.emailCollection = this.options.emailCollection;

            this.loadNotReadCounts();

            this.listenTo(this.emailCollection, 'sync', this.loadNotReadCounts);
            this.listenTo(this.emailCollection, 'folders-update', this.loadNotReadCounts);

            this.listenTo(this.emailCollection, 'all-marked-read', () => {
                this.countsData = this.countsData || {};

                for (let id in this.countsData) {
                    if (id === this.FOLDER_DRAFTS) {
                        continue;
                    }

                    this.countsData[id] = 0;
                }

                this.renderCounts();
            });

            this.listenTo(this.emailCollection, 'draft-sent', () => {
                this.decreaseNotReadCount(this.FOLDER_DRAFTS);
                this.renderCounts();
            });

            this.listenTo(this.emailCollection, 'change:isRead', model => {
                if (this.countsIsBeingLoaded) {
                    return;
                }

                this.manageCountsDataAfterModelChanged(model);
            });

            this.listenTo(this.emailCollection, 'model-removing', id => {
                let model = this.emailCollection.get(id);

                if (!model) {
                    return;
                }

                if (this.countsIsBeingLoaded) {
                    return;
                }

                this.manageModelRemoving(model);
            });

            this.listenTo(this.emailCollection, 'moving-to-trash', (id, model) => {
                model = this.emailCollection.get(id) || model;

                if (!model) {
                    return;
                }

                if (this.countsIsBeingLoaded) {
                    return;
                }

                this.manageModelRemoving(model);
            });

            this.listenTo(this.emailCollection, 'retrieving-from-trash', (id, model) => {
                model = this.emailCollection.get(id) || model;

                if (!model) {
                    return;
                }

                if (this.countsIsBeingLoaded) {
                    return;
                }

                this.manageModelRetrieving(model);
            });
        },

        manageModelRemoving: function (model) {
            if (model.get('status') === 'Draft') {
                this.decreaseNotReadCount(this.FOLDER_DRAFTS);
                this.renderCounts();

                return;
            }

            if (!model.get('isUsers')) {
                return;
            }

            if (model.get('isRead')) {
                return;
            }

            let folderId = model.get('groupFolderId') ?
                ('group:' + model.get('groupFolderId')) :
                (model.get('folderId') || this.FOLDER_INBOX);

            this.decreaseNotReadCount(folderId);
            this.renderCounts();
        },

        manageModelRetrieving: function (model) {
            if (!model.get('isUsers')) {
                return;
            }

            if (model.get('isRead')) {
                return;
            }

            let folderId = model.get('groupFolderId') ?
                ('group:' + model.get('groupFolderId')) :
                (model.get('folderId') || this.FOLDER_INBOX);

            this.increaseNotReadCount(folderId);
            this.renderCounts();
        },

        manageCountsDataAfterModelChanged: function (model) {
            if (!model.get('isUsers')) {
                return;
            }

            let folderId = model.get('groupFolderId') ?
                ('group:' + model.get('groupFolderId')) :
                (model.get('folderId') || this.FOLDER_INBOX);

            !model.get('isRead') ?
                this.increaseNotReadCount(folderId) :
                this.decreaseNotReadCount(folderId);

            this.renderCounts();
        },

        increaseNotReadCount: function (folderId) {
            this.countsData = this.countsData || {};
            this.countsData[folderId] = this.countsData[folderId] || 0;
            this.countsData[folderId]++;
        },

        decreaseNotReadCount: function (folderId) {
            this.countsData = this.countsData || {};

            this.countsData[folderId] = this.countsData[folderId] || 0;

            if (this.countsData[folderId]) {
                this.countsData[folderId]--;
            }
        },

        selectFolder: function (id) {
            this.emailCollection.reset();
            this.emailCollection.abortLastFetch();

            this.selectedFolderId = id;
            this.trigger('select', id);
        },

        afterRender: function () {
            if (this.countsData) {
                this.renderCounts();
            }
        },

        loadNotReadCounts: function () {
            if (this.countsIsBeingLoaded) {
                return;
            }

            this.countsIsBeingLoaded = true;

            Espo.Ajax.getRequest('Email/inbox/notReadCounts').then(data => {
                this.countsData = data;

                if (this.isRendered()) {
                    this.renderCounts();
                    this.countsIsBeingLoaded = false;

                    return;
                }

                this.once('after:render', () => {
                    this.renderCounts();
                    this.countsIsBeingLoaded = false;
                });
            });
        },

        renderCounts: function () {
            let data = this.countsData;

            for (let id in data) {
                let value = '';

                if (data[id]) {
                    value = data[id].toString();
                }

                this.$el.find('li a.count[data-id="'+id+'"]').text(value);
            }
        },
    });
});
